<template>
  <div class="xtx-city">
    <!-- 默认显示的文字 -->
    <div class="select" @click="toggleDialog" :class="{ active }">
      <span class="placeholder" v-if="!seled.fullSels">请选择配送地址</span>
      <!-- 展示选择完的结果 -->
      <span class="value" v-else>{{seled.fullSels}}</span>
      <i class="iconfont icon-angle-down"></i>
    </div>
    <div class="option" v-if="active">
      <!-- 选择城市的弹层  3级联动 省 --市--区 -->
      <span
        class="ellipsis"
        @click="selectCity(city)"
        v-for="city in cityList"
        :key="city.code"
        >{{ city.name }}</span
      >
    </div>
  </div>
</template>

<script>
// 导入vue函数
import { ref, onMounted, reactive } from 'vue'
// 导入axios
import axios from 'axios'

// 获取城市列表数据
const useCityList = () => {
  // 定义一个空数组(接收城市列表)
  const cityList = ref([])
  // 备份数据变量
  const cityListBackup = []
  // 发起请求,获取城市列表数据,并解构出来
  onMounted(async () => {
    const { data } = await axios({
      url: 'https://yjy-oss-files.oss-cn-zhangjiakou.aliyuncs.com/tuxian/area.json'
    })
    console.log('城市列表：', data)
    // 备份数据(这里不能直接写等号赋值,因为是异步的,如果直接等号赋值,就会先执行这里,备份的就是空)
    cityListBackup.push(...data)
    // 赋值
    cityList.value = data
  })
  return { cityList, cityListBackup }
}

export default {
  name: 'XtxCity',
  // 声明子传父事件名(为了系统稳定,不爆错)
  emits: ['change'],

  setup (props, { emit }) {
    // 定义选择层是否显示
    const active = ref(false)
    // 获取城市列表数据,并解构出来
    const { cityList, cityListBackup } = useCityList()
    // 点击选择省市区(核心)
    /**
     * child --> 省(父级,默认显示) --> 市(子) --> 区(子)
     * 选择的数据要不要存?  存完提供给外边用(子传父)
     */
    // 交互功能实现
    // 定义点击选中城市数据
    const seled = reactive({
      provinceCode: '', // 省code
      provinceName: '', // 省名称
      cityCode: '', // 城市code
      cityName: '', // 城市名称
      countryCode: '', // 地区code
      countryName: '', // 地区名
      // 最终选择(省+市+区 -->  拼接为字符串)
      fullSels: ''
    })
    // city -->省(第一次传)>市(第二次传)>区(第三次传)
    const selectCity = (city) => {
      // console.log('child', city.areaList)
      // 第一步:交互
      cityList.value = city.areaList
      // 第二步:存储
      /**
       * 根据获取到的数据中, level=0 是省, level=1 是市级, level=2 是区级,来做if判断+赋值
       */
      if (city.level === 0) {
        // 省
        seled.provinceName = city.name
        seled.provinceCode = city.code
      } else if (city.level === 1) {
        // 市
        seled.cityName = city.name
        seled.cityCode = city.code
      } else {
        // 区
        seled.countryName = city.name
        seled.countryCode = city.code
        // 拼接(最终所有的数据,全部拼接为字符串)
        seled.fullSels = `${seled.provinceName} ${seled.cityName} ${seled.countryName}`
        // 选完了 -->  1.关闭选择框  2.子传父(选择的数据)
        // 1.关闭
        toggleDialog()
        // 2.子传父
        emit('change', seled)
      }
    }

    // 打开下拉菜单
    const open = () => {
      // 点击取反赋值,就是最简单的交互效果,但是这里需分开写,因为open 和 close时,都有需求要做
      // active.value = !active.value
      active.value = true
      // 还原
      console.log('open:', cityListBackup)
      cityList.value = cityListBackup
    }

    // 关闭下拉菜单
    const close = () => {
      active.value = false
    }

    // 点击配置地址的事件
    const toggleDialog = () => {
      //   if判断是,如果状态是true就是打开状态,点击后就要执行 关闭的方法,反之一样
      if (active.value) {
        close()
      } else {
        open()
      }
    }
    return { active, toggleDialog, selectCity, cityList, seled }
  }
}
</script>

<style scoped lang="less">
.xtx-city {
  display: inline-block;
  position: relative;
  z-index: 400;
  margin-left: 10px;
  .select {
    border: 1px solid #e4e4e4;
    height: 30px;
    padding: 0 5px;
    line-height: 28px;
    cursor: pointer;
    &.active {
      background: #fff;
    }
    .placeholder {
      color: #999;
    }
    .value {
      color: #666;
      font-size: 12px;
    }
    i {
      font-size: 12px;
      margin-left: 5px;
    }
  }
  .option {
    width: 542px;
    border: 1px solid #e4e4e4;
    position: absolute;
    left: 0;
    top: 29px;
    background: #fff;
    min-height: 30px;
    line-height: 30px;
    display: flex;
    flex-wrap: wrap;
    padding: 10px;
    > span {
      width: 130px;
      text-align: center;
      cursor: pointer;
      border-radius: 4px;
      padding: 0 3px;
      &:hover {
        background: #f5f5f5;
      }
    }
  }
}
</style>
